home *** CD-ROM | disk | FTP | other *** search
/ Internet Info 1994 March / Internet Info CD-ROM (Walnut Creek) (March 1994).iso / networking / ip / ka9q / alpha.arc / UDP.C < prev    next >
C/C++ Source or Header  |  1988-07-02  |  8KB  |  308 lines

  1. /* Send and receive User Datagram Protocol packets */
  2. #include "global.h"
  3. #include "mbuf.h"
  4. #include "netuser.h"
  5. #include "udp.h"
  6. #include "internet.h"
  7.  
  8. /* Hash table for UDP structures */
  9. struct udp_cb *udps[NUDP] = { NULLUDP} ;
  10. struct udp_stat udp_stat;    /* Statistics */
  11.  
  12. /* Create a UDP control block for lsocket, so that we can queue
  13.  * incoming datagrams.
  14.  */
  15. int
  16. open_udp(lsocket,r_upcall)
  17. struct socket *lsocket;
  18. void (*r_upcall)();
  19. {
  20.     register struct udp_cb *up;
  21.     struct udp_cb *lookup_udp();
  22.     int16 hval,hash_udp();
  23.  
  24.     if((up = lookup_udp(lsocket)) != NULLUDP)
  25.         return 0;    /* Already exists */
  26.     if((up = (struct udp_cb *)malloc(sizeof (struct udp_cb))) == NULLUDP){
  27.         net_error = NO_SPACE;
  28.         return -1;
  29.     }
  30.     up->rcvq = NULLBUF;
  31.     up->rcvcnt = 0;
  32.     up->socket.address = lsocket->address;
  33.     up->socket.port = lsocket->port;
  34.     up->r_upcall = r_upcall;
  35.  
  36.     hval = hash_udp(lsocket);
  37.     up->next = udps[hval];
  38.     up->prev = NULLUDP;
  39.     if(up->next != NULLUDP)
  40.         up->next->prev = up;
  41.     udps[hval] = up;
  42.     return 0;
  43. }
  44.  
  45. /* Send a UDP datagram */
  46. int
  47. send_udp(lsocket,fsocket,tos,ttl,data,length,id,df)
  48. struct socket *lsocket;        /* Source socket */
  49. struct socket *fsocket;        /* Destination socket */
  50. char tos;            /* Type-of-service for IP */
  51. char ttl;            /* Time-to-live for IP */
  52. struct mbuf *data;        /* Data field, if any */
  53. int16 length;            /* Length of data field */
  54. int16 id;            /* Optional ID field for IP */
  55. char df;            /* Don't Fragment flag for IP */
  56. {
  57.     struct mbuf *htonudp(),*bp;
  58.     struct pseudo_header ph;
  59.     struct udp udp;
  60.  
  61.     length = UDPHDR;
  62.     if(data != NULLBUF)
  63.         length += len_mbuf(data);
  64.  
  65.     udp.source = lsocket->port;
  66.     udp.dest = fsocket->port;
  67.     udp.length = length;
  68.  
  69.     /* Create IP pseudo-header, compute checksum and send it */
  70.     ph.length = length;
  71.     ph.source = lsocket->address;
  72.     ph.dest = fsocket->address;
  73.     ph.protocol = UDP_PTCL;
  74.  
  75.     if((bp = htonudp(&udp,data,&ph)) == NULLBUF){
  76.         net_error = NO_SPACE;
  77.         free_p(data);
  78.         return 0;
  79.     }
  80.     udp_stat.sent++;
  81.     ip_send(lsocket->address,fsocket->address,UDP_PTCL,tos,ttl,bp,length,id,df);
  82.     return length;
  83. }
  84.  
  85. /* Accept a waiting datagram, if available. Returns length of datagram */
  86. int
  87. recv_udp(lsocket,fsocket,bp)
  88. struct socket *lsocket;        /* Local socket to receive on */
  89. struct socket *fsocket;        /* Place to stash incoming socket */
  90. struct mbuf **bp;            /* Place to stash data packet */
  91. {
  92.     struct udp_cb *lookup_udp();
  93.     register struct udp_cb *up;
  94.     struct socket *sp;
  95.     struct mbuf *buf;
  96.     int16 length;
  97.  
  98.     up = lookup_udp(lsocket);
  99.     if(up == NULLUDP){
  100.         net_error = NO_CONN;
  101.         return -1;
  102.     }
  103.     if(up->rcvcnt == 0){
  104.         net_error = WOULDBLK;
  105.         return -1;
  106.     }
  107.     buf = dequeue(&up->rcvq);
  108.     up->rcvcnt--;
  109.  
  110.     sp = (struct socket *)buf->data;
  111.     /* Fill in the user's foreign socket structure, if given */
  112.     if(fsocket != NULLSOCK){
  113.         fsocket->address = sp->address;
  114.         fsocket->port = sp->port;
  115.     }
  116.     /* Strip socket header and hand data to user */
  117.     pullup(&buf,NULLCHAR,sizeof(struct socket));
  118.     length = len_mbuf(buf);
  119.     if(bp != (struct mbuf **)NULL)
  120.         *bp = buf;
  121.     else
  122.         free_p(buf);
  123.     return length;
  124. }
  125. /* Delete a UDP control block */
  126. int
  127. del_udp(lsocket)
  128. struct socket *lsocket;
  129. {
  130.     register struct udp_cb *up;
  131.     struct udp_cb *lookup_udp();
  132.     struct mbuf *bp;
  133.     int16 hval,hash_udp();
  134.  
  135.     if((up = lookup_udp(lsocket)) == NULLUDP){
  136.         net_error = INVALID;
  137.         return -1;
  138.     }        
  139.     /* Get rid of any pending packets */
  140.     while(up->rcvcnt != 0){
  141.         bp = up->rcvq;
  142.         up->rcvq = up->rcvq->anext;
  143.         free_p(bp);
  144.         up->rcvcnt--;
  145.     }
  146.     hval = hash_udp(&up->socket);
  147.     if(udps[hval] == up){
  148.         /* First on list */
  149.         udps[hval] = up->next;
  150.         up->next->prev = NULLUDP;
  151.     } else {
  152.         up->prev->next = up->next;
  153.         up->next->prev = up->prev;
  154.     }
  155.     free((char *)up);
  156.     return 0;
  157. }
  158. /* Process an incoming UDP datagram */
  159. void
  160. udp_input(bp,protocol,source,dest,tos,length,rxbroadcast)
  161. struct mbuf *bp;    /* UDP header and data */
  162. char protocol;        /* Should always be 17 */
  163. int32 source;        /* Source IP address */
  164. int32 dest;        /* Dest IP address */
  165. char tos;
  166. int16 length;
  167. char rxbroadcast;    /* The only protocol that accepts 'em */
  168. {
  169.     struct pseudo_header ph;
  170.     struct udp udp;
  171.     struct udp_cb *up,*lookup_udp();
  172.     struct socket lsocket;
  173.     struct socket *fsocket;
  174.     struct mbuf *packet;
  175.     int ckfail = 0;
  176.  
  177.     if(bp == NULLBUF)
  178.         return;
  179.  
  180.     udp_stat.rcvd++;
  181.  
  182.     /* Create pseudo-header and verify checksum */
  183.     ph.source = source;
  184.     ph.dest = dest;
  185.     ph.protocol = protocol;
  186.     ph.length = length;
  187.  
  188.     if(cksum(&ph,bp,length) != 0)
  189.         /* Checksum apparently failed, note for later */
  190.         ckfail++;
  191.  
  192.     /* Extract UDP header in host order */
  193.     ntohudp(&udp,&bp);
  194.  
  195.     /* If the checksum field is zero, then ignore a checksum error.
  196.      * I think this is dangerously wrong, but it is in the spec.
  197.      */
  198.     if(ckfail && udp.checksum != 0){
  199.         udp_stat.cksum++;
  200.         free_p(bp);
  201.         return;
  202.     }
  203.     /* If this was a broadcast packet, pretend it was sent to us */
  204.     if(rxbroadcast){
  205.         lsocket.address = ip_addr;
  206.         udp_stat.bdcsts++;
  207.     } else
  208.         lsocket.address = dest;
  209.  
  210.     lsocket.port = udp.dest;
  211.     /* See if there's somebody around to read it */
  212.     if((up = lookup_udp(&lsocket)) == NULLUDP){
  213.         /* Nope, toss it on the floor */
  214.         udp_stat.unknown++;
  215.         free_p(bp);
  216.         return;
  217.     }
  218.     /* Create space for the foreign socket info */
  219.     if((packet = pushdown(bp,sizeof(struct socket))) == NULLBUF){
  220.         /* No space, drop whole packet */
  221.         free_p(bp);
  222.         return;
  223.     }
  224.     fsocket = (struct socket *)packet->data;
  225.     fsocket->address = source;
  226.     fsocket->port = udp.source;
  227.  
  228.     /* Queue it */
  229.     enqueue(&up->rcvq,packet);
  230.     up->rcvcnt++;
  231.     if(up->r_upcall)
  232.         (*up->r_upcall)(&lsocket,up->rcvcnt);
  233. }
  234. /* Look up UDP socket, return control block pointer or NULLUDP if nonexistant */
  235. static
  236. struct udp_cb *
  237. lookup_udp(socket)
  238. struct socket *socket;
  239. {
  240.     register struct udp_cb *up;
  241.     int16 hash_udp();
  242.  
  243.     up = udps[hash_udp(socket)];
  244.     while(up != NULLUDP){
  245.         if(memcmp((char *)socket,(char *)&up->socket,sizeof(struct socket)) == 0)
  246.             break;
  247.         up = up->next;
  248.     }
  249.     return up;
  250. }
  251.  
  252. /* Hash a UDP socket (address and port) structure */
  253. static
  254. int16
  255. hash_udp(socket)
  256. struct socket *socket;
  257. {
  258.     int16 hval;
  259.  
  260.     /* Compute hash function on socket structure */
  261.     hval = hiword(socket->address);
  262.     hval ^= loword(socket->address);
  263.     hval ^= socket->port;
  264.     hval %= NUDP;
  265.     return hval;
  266. }
  267. /* Convert UDP header in internal format to an mbuf in external format */
  268. struct mbuf *
  269. htonudp(udp,data,ph)
  270. struct udp *udp;
  271. struct mbuf *data;
  272. struct pseudo_header *ph;
  273. {
  274.     struct mbuf *bp;
  275.     register char *cp;
  276.     int16 checksum;
  277.  
  278.     /* Allocate UDP protocol header and fill it in */
  279.     if((bp = pushdown(data,UDPHDR)) == NULLBUF)
  280.         return NULLBUF;
  281.  
  282.     cp = bp->data;
  283.     cp = put16(cp,udp->source);    /* Source port */
  284.     cp = put16(cp,udp->dest);    /* Destination port */
  285.     cp = put16(cp,udp->length);    /* Length */
  286.     *cp++ = 0;            /* Clear checksum */
  287.     *cp-- = 0;
  288.  
  289.     /* All zeros and all ones is equivalent in one's complement arithmetic;
  290.      * the spec requires us to change zeros into ones to distinguish an
  291.       * all-zero checksum from no checksum at all
  292.      */
  293.     if((checksum = cksum(ph,bp,ph->length)) == 0)
  294.         checksum = 0xffffffff;
  295.     put16(cp,checksum);
  296.     return bp;
  297. }
  298. /* Convert UDP header in mbuf to internal structure */
  299. ntohudp(udp,bpp)
  300. struct udp *udp;
  301. struct mbuf **bpp;
  302. {
  303.     udp->source = pull16(bpp);
  304.     udp->dest = pull16(bpp);
  305.     udp->length = pull16(bpp);
  306.     udp->checksum = pull16(bpp);
  307. }
  308.